

#### INTERRUPT ED ECCEZIONI

I. Frosio

#### SOMMARIO

- Eccezioni ed interrupt Esempi
- o Gestione SW delle eccezioni: i dettagli



#### **ECCEZIONI ED INTERRUPT**

- <u>Eccezione</u>: interna al processore (es. overflow), modifica il flusso di esecuzione di un'istruzione.
- Interrupt: esterno al processore (es. click del mouse), viene generalmente attesa la fine del ciclo dell'istruzione prima di servire l'interrupt.

| Tipo di evento                                  | Provenienza | Terminologia MIPS           |
|-------------------------------------------------|-------------|-----------------------------|
| Richiesta di un dispositivo di I/O              | Esterna     | Interrupt                   |
| Chiamata al SO da parte di un programma Interna |             | Eccezione                   |
| Overflow aritmetico                             | Interna     | Eccezione                   |
| Uso di un'istruzione non definita               | Interna     | Eccezione                   |
| Malfunzionamento dell'hardware                  | Entrambe    | Eccezione o<br>Interruzione |

#### RISPOSTA ALL'ECCEZIONE

- Interviene il sistema operativo.
- Risposta vettorializzata: salto ad un indirizzo diverso per ogni diverso tipo di eccezione.
- Tramite registro (causa): un unico entry point (0x80000180 in MIPS) per la gestione delle eccezioni; la causa dell'eccezione è nel registro causa.
- Valore del registro causa: 0x00000000 se non ci sono eccezioni, altrimenti contiene il codice dell'eccezione.
- NB: l'HW aggiorna automaticamente il registro causa al verificarsi dell'eccezione. La gestione dell'eccezione è SW (coordinamento HW / SW).

#### REGISTRO CAUSA IN PC-SPIM PCSpim File Simulator Window Help EPC BadVAddr= 00000000 = 00000000 = 00000000Cause = 00000000 Status = 3000ff10 = 00000000 LO = 000000000General Registers R0 (r0) = 00000000 R8 (t0) = 00000000 R16 (s0) = 00000000 R24 (t8) = 00000000(at) = 00000000 R9 (t1) = 00000000 R17 (s1) = 00000000 R25 (t9) = 00000000(v0) = 00000000 R10 (t2) = 00000000 R18 (s2) = 00000000 R26 (k0) = 00000000(v1) = 00000000 R11 (t3) = 00000000 R19 (s3) = 00000000 R27 (k1) = 00000000(a0) = 00000000 R12 (t4) = 00000000 R20 (s4) = 00000000 R28 (qp) = 10008000[0x00400000] 0x8fa40000 lw \$4, 0(\$29) ; 183: lw \$a0 0(\$sp) # argc 0x27a50004 addiu \$5, \$29, 4 [0x00400004] ; 184: addiu \$a1 \$sp 4 # argv 0x24a60004 addiu \$6, \$5, 4 ; 185: addiu \$a2 \$a1 4 # envp [0x00400008] [0x0040000cl 0x00041080 sll \$2, \$4, 2 ; 186: sll \$v0 \$a0 2 0x00c23021 addu \$6, \$6, \$2 : 187: addu \$a2 \$a2 \$v0 [0x004000101 0x0c000000 jal 0x00000000 [main] ; 188: jal main [0x00400014] qon 00000000x0 [0x00400018] : 189: nop 0x3402000a ori \$2, \$0, 10 ; 191: li \$v0 10 [0x0040001c] DATA [0x10000000]...[0x10040000] 0x00000000 STACK [0x7ffff4f4] 0x00000000 0x00000000 0x7fffffe1 KERNEL DATA [0x900000001 0x78452020 0x74706563 0x206e6f69 0x636f2000 SPIM Version 8.0 of January 8, 2010 Copyright 1990-2010, James R. Larus. All Rights Reserved. DOS and Windows ports by David A. Carley. Copyright 1997, Morgan Kaufmann Publishers, Inc. See the file README for a full copyright notice. Loaded: C:\Program Files\PCSpim\exceptions.s PC=0x00000000 EPC=0x00000000 Cause=0x00000000 DELAY LD For Help, press F1 10:54 AM 5/5/2010

### ECCEZIONE: ESEMPIO PC-SPIM (Es. 5.1)

Si esegua il seguente programma:

o main:

• lui \$t0, 0xffff
# \$t0=0xffff0000

o ori \$t0, \$t0, 0xffff # \$t0=0xffffffff

o jr \$t0 # jump to 0xffffffff (exception)

o addi \$v0, \$zero, 10

syscall

Si analizzi il comportamento del processore MIPS alla chiamata di jr.

#### ESERCIZIO 5.1 - SOLUZIONE

- Il processore salta a 0xffffffff e carica l'istruzione di questa locazione di memoria;
- Non viene eseguita la fetch(l'indirizzo non contiene infatti alcuna istruzione) -> eccezione!
- Nel registro causa compare il codice di eccezione 0x0000018 (istruzione non può essere decodificata);
- Nel registro EPC compare l'indirizzo 0xffffffc, per un eventuale ritorno alla riga di codice che ha generato l'eccezione. Questo è l'indirizzo della word che ha generato l'errore (così come supposto dall'HW).
- Exception 6 -> bus error on instruction fetch.



### ECCEZIONE: ESEMPIO PC-SPIM (Es. 5.1)

- Non è possibile effettuare jump a 0xfffffff...
- Nel registro EPC viene salvato l'indirizzo dell'istruzione che ha generato l'interruzione (in questo caso 0xffffffc = 0xfffffff-3), nel caso in cui sia possibile un ritorno dall'eccezione alla normale esecuzione del codice.
- Il "rientro dall'eccezione porta ad una riga di codice non valida"! La procedura di gestione dell'eccezione termina quindi l'esecuzione del programma.

#### ECCEZIONE: ESERCIZIO Es. 5.2

o main:

lui \$t0, 0x8000

lui \$t1, 0x8000

add \$t2, \$t0, \$t1

o addi \$t3, \$t2, -100

# \$t0=0x80000000

# \$t1=0x80000000

# Overflow

# Next instruction

- o addi \$v0, \$zero, 10
- syscall
- Si genera un overflow. Dopo avere gestito l'eccezione (info sulla console per l'utente), il programma riprende il suo flusso normale.
- Si analizzi il comportamento del registro cause e la gestione dell'eccezione da parte del sistema operativo.

#### Es. 5.2 - Osservazioni

- Nel registro cause compare il codice 0x00000030 -> overflow.
- C'è una salto all'indirizzo di ritorno, effettuato tenendo conto del valore di EPC, che fa continuare il flusso del programma.

#### SOMMARIO

- Eccezioni ed interrupt Esempi
- o Gestione SW delle eccezioni: i dettagli

#### GESTIONE SW DI UN'ECCEZIONE

- L'HW rileva un'eccezione;
- L'HW aggiorna il contenuto del registro EPC (Exception Program Counter) con l'indirizzo dell'istruzione che ha causato l'eccezione;
- L'HW aggiorna il registro Cause con un codice che specifichi il motivo per il quale si è verificata l'eccezione;

O ...

- Cosa sono i registro EPC e Cause?
- I registri EPC e Cause fanno parte, con altri registri, del cosidetto coprocessore 0...

#### REGISTRI DEL COPROCESSORE 0

Un insieme di registri speciali, denominati coprocessore 0, per la gestione (tra le altre cose) delle eccezioni.

| Nome del<br>registro | Numero del<br>registro in<br>coprocessore 0 | Utilizzo                                                                                                   |
|----------------------|---------------------------------------------|------------------------------------------------------------------------------------------------------------|
| Bad/Addr             | 8                                           | Registro contenente l'indirizzo di memoria a cui si è fatto riferimento (cf. "page fault").                |
| Count                | 9                                           | Timer (MIPS: 10ms).                                                                                        |
| Compare              | 11                                          | Valore da comparare con un timer. Genera un interrupt.                                                     |
| Status               | 12                                          | Maschera delle interruzioni e bit di abilitazione. Stato dei<br>diversi livelli di priorità (6 HW e 2 SW). |
| Cause                | 13                                          | Tipo dell'interruzione e bit delle interruzioni pendenti                                                   |
| EPC                  | 14                                          | Registro contenente l'indirizzo dell'istruzione che ha causato l'interruzione.                             |

#### GESTIONE SW DI UN'ECCEZIONE

- L'HW rileva un'eccezione;
- L'HW aggiorna il contenuto del registro EPC (Exception Program Counter) con l'indirizzo dell'istruzione che ha causato l'eccezione;
- L'HW aggiorna il registro Cause con un codice che specifichi il motivo per il quale si è verificata l'eccezione;
- L'HW aggiorna il contenuto del registro PC al valore 0x80000180 (exception handler); qui è presente una procedura del sistema operativo per la gestione delle eccezioni;
- Gestione dell'eccezione da parte dell'OS
- Ritorno all'istruzione EPC+4...
- ... Come è possibile effettuare il ritorno all'istruzione EPC+4?
  - E' necessario utilizzare un jr <r>, ove r è un registro contenente l'indirizzo EPC+4...
  - Come è possibile copiare il contenuto di EPC in un registro generico?

### LETTURA / SCRITTURA DEI REGISTRI DEL CORPOCESSORE 0

- mfc0 \$t0, \$13
   del coprocessore 0 (Cause register) al registro \$t0
- mtc0 \$14, \$t0 # copia dal registro \$t0 al registro \$14 (EPC) del coprocessore 0
- lwc0 \$13, 100(\$t3) # carica una word dall'indirizzo 100(\$t3) al registro \$13 del coprocessore 0
- swc0 \$13, 50(\$t2) # salva la word nel registro \$13 del corpocessore 0 in memoria

#### GESTIONE SW DI UN'ECCEZIONE

- L'HW rileva un'eccezione;
- L'HW aggiorna il contenuto del registro EPC (Exception Program Counter) con l'indirizzo dell'istruzione che ha causato l'eccezione;
- L'HW aggiorna il registro Cause con un codice che specifichi il motivo per il quale si è verificata l'eccezione;
- L'HW aggiorna il contenuto del registro PC al valore 0x80000180 (exception handler); qui è presente una procedura del sistema operativo per la gestione delle eccezioni;
- Gestione dell'eccezione da parte dell'OS
- Ritorno all'istruzione EPC+4...
- ... Come è possibile effettuare il ritorno all'istruzione EPC+4?
  - Copio il contenuto di EPC in un registro (e aggiungo 4) quale indirizzo di ritorno...
  - mfc0 \$t0, \$14 # \$t0 = EPC
  - addi \$t0, \$t0, 4 # \$t0 +=4

#### GESTIONE SW DI UN'ECCEZIONE

- ... Come è possibile effettuare il ritorno all'istruzione EPC+4?
  - Copio il contenuto di EPC in un registro (e aggiungo 4) quale indirizzo di ritorno...
  - mfc0 \$t0, \$14 # \$t0 = EPC
  - addi \$t0, \$t0, 4 # \$t0 +=4
- Ma in questo modo alteriamo il registro \$t0...
- Non possiamo utilizzare lo stack per salvare \$t0 perchè lo stack potrebbe essere corrotto...
- Il registro \$ra non è a disposizione...
- Sono necessari dei registri dedicati alla gestione delle eccezioni!

#### REGISTRI PER LA GESTIONE DELLE ECCEZIONI

```
16 s0 callee saves
   zero constant 0
   at reserved for assembler
                                   ... (caller can clobber)
   v0 expression evaluation &
                                  23 s7
       function results
                                           temporary (cont'd)
                                   24
                                      t8
   a0 arguments
                                   25
                                      t9
   a1
                                  26 k0 reserved for OS kernel
   a2
                                  27: k1
                                      gp Pointer to global area
   a3
       temporary: caller saves
                                          Stack pointer
       (callee can clobber)
                                      fp frame pointer (s8)
                                  31 ra Return Address (HW)
15 t7
```

#### GESTIONE SW DI UN'ECCEZIONE

- L'HW rileva un'eccezione;
- L'HW aggiorna il contenuto del registro EPC (Exception Program Counter) con l'indirizzo dell'istruzione che ha causato l'eccezione;
- L'HW aggiorna il registro Cause con un codice che specifichi il motivo per il quale si è verificata l'eccezione;
- L'HW aggiorna il contenuto del registro PC al valore 0x80000180 (exception handler); qui è presente una procedura del sistema operativo per la gestione delle eccezioni;
- Gestione dell'eccezione da parte dell'OS
- Ritorno all'istruzione EPC+4...
- ... Come è possibile effettuare il ritorno all'istruzione EPC+4?
  - Copio il contenuto di EPC in un registro (e aggiungo 4) quale indirizzo di ritorno...
  - mfc0 \$k0, \$14 # \$k0 = EPC
  - addi \$k0, \$k0, 4 # \$k0 +=4

#### RISPOSTA AD UN'ECCEZIONE

- 1) Prologo
- 2) Risposta all'eccezione (exception handler)
- o 3) Epilogo

#### RISPOSTA AD UN'ECCEZIONE: PROLOGO

- 1. salvataggio in EPC del PC corrente (HW);
- 2. caricamento nei bit 2-5 del registro causa, \$13, del codice dell'eccezione vericatasi (HW);
- 3. salvataggio dello stato;
- 4. caricamento nel PC del valore 0x80000180 (HW).

Non posso salvare lo stato (il contenuto dei registri del register file) in stack (perchè lo stack potrebbe essere causa dell'eccezione), salvo perciò in \$k0, \$k1 e in .kdata. i valori necessari.

Il segmento di dati .kdata svolge il ruolo dello stack per la procedura di risposta ad interrupt ed eccezioni.

#### RISPOSTA AD UN'ECCEZIONE: EPILOGO

- 1. Pulire il cause register e salvarlo;
- 2. Settare a 0 il bit di interrupt corrispondente e salvare lo status register;
- 3. Ripristino dello stato;
- 4. Caricamento nel PC del valore EPC + 4.

#### EXPECPTION HANDLER: PC-SPIM (1)

0x80000180:

move \$k1 \$at # Save \$at # Not re-entrant and we can't trust \$sp nel segmento o sw \$v0 s1 sw \$a0 s2 # But we need to use these registers

Salva \$at, \$v0♀<mark>़\$a0</mark> .kdata

mfc0 \$k0 \$13

# Cause register

Cause in \$k0

o srl \$a0 \$k0 2

# Extract ExcCode Field

andi \$a0 \$a0 0x1f

li \$v0 4

# syscall 4 (print\_str)

la \$a0 m1

syscall

li \$v0 1

# syscall 1 (print\_int)

srl \$a0 \$k0 2

# Extract ExcCode Field

andi \$a0 \$a0 0x1f

Syscall

li \$v0 4

# syscall 4 (print\_str)

srl -> Shift right logical by a constant number of bits

Stampa info per l'utente

### EXPECPTION HANDLER: PC-SPIM (CONT. 2)

- andi \$a0 \$k0 0x3c
- lw \$a0 \_\_excp(\$a0)
- nop
- syscall

#### Stampa info per l'utente

Se PC non è corrotto

- bne \$k0 0x18 ok pc # Bad PC exception requires special checks
- Nop
- mfc0 \$a0 \$14

# EPC \$a0 = EPC

- andi \$a0 \$a0 0x3 # Is EPC word-aligned?
- beq \$a0 0 ok\_pc

Se EPC non è corrotto

- nop
- li \$v0 10

# Exit on really bad PC

- syscall
- o ok\_pc: li \$v0 4

# syscall 4 (print\_str)

- la \$a0 m2
- syscall

Gestione "speciale" per (E)PC corrotto [Exit]

Stampa info per l'utente

26

#### EXPECPTION HANDLER: PC-SPIM

o srl \$a0 \$k0 2

# Extract ExcCode Field

\$a0 = Exception code

o andi \$a0 \$a0 0x1f

o bne \$a0 0 ret

# 0 means exception was an interrupt

nop

o ret: mfc0 \$k0 \$14

# Bump EPC register

o addiu \$k0 \$k0 4

# Skip faulting instruction

o mtc0 \$k0 \$14

lw \$v0 s1

# Restore other registers

lw \$a0 s2

o move \$at \$k1

# Restore \$at

mtc0 \$0 \$13

# Clear Cause register

mfc0 \$k0 \$12

# Set Status register

ori \$k0 0x1

# Interrupts enabled

mtc0 \$k0 \$12

Se è un interrupt...

k0 = EPC + 4

EPC = \$k0

Ripristina valori originali dei registri

Clear

eret

**Ritorno** 

27

#### RISPOSTA AD UN'ECCEZIONE: PROLOGO

- Exception handler (0x80000180):
  - stampa un messaggio solo se si verifica un'eccezione.
  - Il messaggio conterrà l'indirizzo (in \$a0) ed il codice dell'eccezione (in \$a1).
  - Non fa nulla se si verifica un interrupt.
- Occorrerà identificare se si tratta di un'eccezione (allo scopo controllo se i bit che identificano le eccezioni sono tutti a 0 nel registro causa).
- Inserirò quindi in \$a0 il codice dell'eccezione ed in \$a1 l'indirizzo (dovrò quindi prima salvare il contenuto di \$a0 e \$a1 in .kdata).
- Per il corpo della procedura dovrò copiare i registri da coprocessore 0 a register file per poterne elaborare il contenuto.

#### .ktext 0x80000180

# Prologo della procedura, inizia dall'entry point

```
la $k0, save0  # Handler is not re-entrant and can't use
sw $a0, 0($k0)  # stack to save $a0, $a1 (data required are
```

sw \$a1, 4(\$k0) # stored in kernel data segment)

#### .kdata

save0: .word 0, 0

\$k0, \$k1 (e \$at) sono considerati registri temporanei riservati al

SO. Non devono quindi essere salvati in .kdata

### RISPOSTA AD UN'ECCEZIONE: CORPO DELLA PROCEDURA

# Corpo della procedura

o mfc0 \$k0, \$13

o srl \$k0, \$k0, 2

andi \$a0, \$k0, 0x1f

beq \$a0, \$zero, epilogo

o mfc0 \$k0, \$12

andi \$k0, \$k0, 0xfffe

ori \$k0, \$k0, 0x2

mtco \$12, \$k0

o mfc0 \$a1, \$14

jal print\_excp

# Move Cause into \$k0 of register file

# Prepare ExcepCode field at position 0

# Extract ExcCode field (5 bits) and prepare it for printing (insert in a0)

# Branch if ExcepCode=0: not an exception

# Status register into \$k0

# Mask all interrupts

# Interrupt at exception level

# Move EPC into \$a1

# Print exception error message.

# Parameters are in \$a0, \$a1

- Quando arriva un interrupt, imposta il suo bit di interrupt corrispondente nel registro Causa, anche nello status register il bit corrispondente nella maschera è disabilitato.
- Quando un interrupt è pending, interromperà il processore, quando il suo bit corrispondente nella sua maschera nello status register è stato abilitato.

#### RISPOSTA AD UN'ECCEZIONE: EPILOGO

```
mtc0 $13, $0
                              # Clear Cause register
        mfc0 $k0, $12
                              # To fix Status register (bring status in k0)
        andi $k0, 0xfffd
                              # Clear EXL bit (Exception level, mask bit)
                              # in status register
                              # Enable interrupts in status register
        ori $k0, 0x1
        mtc0 $12, $k0
                              # Restore status register
Epilogo: mfc0 $k1, $14
                              # Bump EPC into $k0
        addiu $k0, $k1, 4
                              # Do not reexecute faulting instruction:
                              # EPC = EPC + 4
        mtc0 $14, $k1
                               # Write EPC + 4 in EPC
        la $k0, save0
        lw $a0, 0($k0)
                              # Restore previously saved registers
        lw $a1, 4($k0)
        jr $k1
                              # Return to EPC address (eret)
                     15
                                                                Status register
```

30

#### ESERCIZIO 5.4

- Si analizzi il codice exception.s;
- Si identifichi la porzione di codice ove è possibile inserire la gestione degli interrupt (es. ctrl+C).